# -*- coding: utf-8 -*-
# 版权所有 2019 深圳米筐科技有限公司（下称“米筐科技”）
#
# 除非遵守当前许可，否则不得使用本软件。
#
#     * 非商业用途（非商业用途指个人出于非商业目的使用本软件，或者高校、研究所等非营利机构出于教育、科研等目的使用本软件）：
#         遵守 Apache License 2.0（下称“Apache 2.0 许可”），
#         您可以在以下位置获得 Apache 2.0 许可的副本：http://www.apache.org/licenses/LICENSE-2.0。
#         除非法律有要求或以书面形式达成协议，否则本软件分发时需保持当前许可“原样”不变，且不得附加任何条件。
#
#     * 商业用途（商业用途指个人出于任何商业目的使用本软件，或者法人或其他组织出于任何目的使用本软件）：
#         未经米筐科技授权，任何个人不得出于任何商业目的使用本软件（包括但不限于向第三方提供、销售、出租、出借、转让本软件、
#         本软件的衍生产品、引用或借鉴了本软件功能或源代码的产品或服务），任何法人或其他组织不得出于任何目的使用本软件，
#         否则米筐科技有权追究相应的知识产权侵权责任。
#         在此前提下，对本软件的使用同样需要遵守 Apache 2.0 许可，Apache 2.0 许可与本许可冲突之处，以本许可为准。
#         详细的授权流程，请联系 public@ricequant.com 获取。

from typing import Optional

from rqalpha.interface import AbstractFrontendValidator, AbstractPosition
from rqalpha.const import POSITION_EFFECT
from rqalpha.utils.logger import user_system_log
from rqalpha.model.order import Order
from rqalpha.portfolio.account import Account

from rqalpha.utils.i18n import gettext as _


class PositionValidator(AbstractFrontendValidator):
    def validate_cancellation(self, order: Order, account: Optional[Account] = None) -> Optional[str]:
        return None
    
    def validate_submission(self, order: Order, account: Optional[Account] = None) -> Optional[str]:
        if account is None:
            return None
        if order.position_effect in (POSITION_EFFECT.OPEN, POSITION_EFFECT.EXERCISE):
            return None
        position = account.get_position(order.order_book_id, order.position_direction)  # type: AbstractPosition
        if order.position_effect == POSITION_EFFECT.CLOSE_TODAY and order.quantity > position.today_closable:
            reason = _(
                "Order Creation Failed: not enough today position {order_book_id} to close, target"
                " quantity is {quantity}, closable today quantity is {closable}").format(
                order_book_id=order.order_book_id,
                quantity=order.quantity,
                closable=position.today_closable,
            )
            return reason
        if order.position_effect == POSITION_EFFECT.CLOSE and order.quantity > position.closable:
            reason = _(
                "Order Creation Failed: not enough position {order_book_id} to close or exercise, target"
                " sell quantity is {quantity}, closable quantity is {closable}").format(
                order_book_id=order.order_book_id,
                quantity=order.quantity,
                closable=position.closable,
            )
            return reason
        return None
